/*
 * @lc app=leetcode.cn id=904 lang=cpp
 *
 * [904] 水果成篮
 *
 * https://leetcode.cn/problems/fruit-into-baskets/description/
 *
 * algorithms
 * Medium (43.35%)
 * Likes:    337
 * Dislikes: 0
 * Total Accepted:    88.3K
 * Total Submissions: 197K
 * Testcase Example:  '[1,2,1]'
 *
 * 你正在探访一家农场，农场从左到右种植了一排果树。这些树用一个整数数组 fruits 表示，其中 fruits[i] 是第 i 棵树上的水果
 * 种类 。
 *
 * 你想要尽可能多地收集水果。然而，农场的主人设定了一些严格的规矩，你必须按照要求采摘水果：
 *
 *
 * 你只有 两个 篮子，并且每个篮子只能装 单一类型 的水果。每个篮子能够装的水果总量没有限制。
 * 你可以选择任意一棵树开始采摘，你必须从 每棵 树（包括开始采摘的树）上 恰好摘一个水果
 * 。采摘的水果应当符合篮子中的水果类型。每采摘一次，你将会向右移动到下一棵树，并继续采摘。
 * 一旦你走到某棵树前，但水果不符合篮子的水果类型，那么就必须停止采摘。
 *
 *
 * 给你一个整数数组 fruits ，返回你可以收集的水果的 最大 数目。
 *
 *
 *
 * 示例 1：
 *
 *
 * 输入：fruits = [1,2,1]
 * 输出：3
 * 解释：可以采摘全部 3 棵树。
 *
 *
 * 示例 2：
 *
 *
 * 输入：fruits = [0,1,2,2]
 * 输出：3
 * 解释：可以采摘 [1,2,2] 这三棵树。
 * 如果从第一棵树开始采摘，则只能采摘 [0,1] 这两棵树。
 *
 *
 * 示例 3：
 *
 *
 * 输入：fruits = [1,2,3,2,2]
 * 输出：4
 * 解释：可以采摘 [2,3,2,2] 这四棵树。
 * 如果从第一棵树开始采摘，则只能采摘 [1,2] 这两棵树。
 *
 *
 * 示例 4：
 *
 *
 * 输入：fruits = [3,3,3,1,2,1,1,2,3,3,4]
 * 输出：5
 * 解释：可以采摘 [1,2,1,1,2] 这五棵树。
 *
 *
 *
 *
 * 提示：
 *
 *
 * 1 <= fruits.length <= 10^5
 * 0 <= fruits[i] < fruits.length
 *
 *
 */

// @lc code=start
#include <iostream>
#include <unordered_set>
#include <vector>
using namespace std;
class Solution {
public:
    vector<pair<int, int>> freq;

    void freqSetup(vector<int> &fruits) {
        for (auto &fruit : fruits) {
            if (freq.empty() || fruit != freq.back().first) {
                freq.push_back({fruit, 1});
            }
            else {
                ++freq.back().second;
            }
        }
        cout << "freqSetup Success" << endl;
        cout << "freqSetup Element:" << endl;
        for (auto &item : freq) {
            cout << item.first << ", " << item.second << endl;
        }
        return;
    }

    int totalFruit(vector<int> &fruits) {
        freqSetup(fruits);
        int n = freq.size();
        if (n == 1) {
            return fruits.size();
        }
        int ans = 0;
        int temp = 0;
        int index = 0;
        unordered_set<int> hash;
        while (index < n) {
            if (hash.size() < 2) {
                hash.insert(freq[index].first);
                temp += freq[index].second;
                ++index;
            }
            else {
                if (hash.count(freq[index].first)) {
                    temp += freq[index].second;
                    ++index;
                }
                else {
                    ans = max(ans, temp);
                    hash.clear();
                    temp = 0;
                    --index;
                }
            }
        }
        ans = max(ans, temp);
        return ans;
    }
};
// @lc code=end
